from info import db
from info.modules.user import user_blu
from flask import render_template, g, abort, request, jsonify, current_app, redirect, url_for
from info.utils.common import user_login_data, upload_file

# 显示个人中心
from info.utils.constants import USER_COLLECTION_MAX_NEWS, QINIU_DOMIN_PREFIX, USER_FOLLOWED_MAX_COUNT
from info.utils.models import UserCollection, Category, News
from info.utils.response_code import RET, error_map


@user_blu.route('/user_info')
@user_login_data
def user_info():
    # 判断用户是否登录
    user = g.user
    if not user:  # 用户未登录, 重定向到前台首页
        return redirect(url_for("home.index"))

    return render_template("news/user.html", user=user.to_dict())


# 显示/修改基本资料
@user_blu.route('/base_info', methods=['GET', 'POST'])
@user_login_data
def base_info():
    # 判断用户是否登录
    user = g.user
    if not user:  # 用户未登录
        return abort(403)

    if request.method == 'GET':  # 展示页面
        return render_template("news/user_base_info.html", user=user.to_dict())

    # POST 提交数据
    # 获取参数
    signature = request.json.get("signature")
    nick_name = request.json.get("nick_name")
    gender = request.json.get("gender")
    # 校验参数
    if not all([signature, nick_name, gender]):
        return jsonify(errno=RET.PARAMERR, errmsg=error_map[RET.PARAMERR])
    if gender not in ['MAN', 'WOMAN']:
        return jsonify(errno=RET.PARAMERR, errmsg=error_map[RET.PARAMERR])

    # 数据库行为: 修改用户数据
    user.signature = signature
    user.nick_name = nick_name
    user.gender = gender

    # json返回结果
    return jsonify(errno=RET.OK, errmsg=error_map[RET.OK])


# 显示/修改头像
@user_blu.route('/pic_info', methods=['GET', 'POST'])
@user_login_data
def pic_info():
    # 判断用户是否登录
    user = g.user
    if not user:  # 用户未登录
        return abort(403)

    if request.method == 'GET':  # 展示页面
        return render_template("news/user_pic_info.html", user=user.to_dict())

    # POST 提交数据
    # 获取参数
    avatar_file = request.files.get("avatar")
    # 读取上传文件的二进制数据
    try:
        img_bytes = avatar_file.read()

    except BaseException as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.PARAMERR, errmsg=error_map[RET.PARAMERR])

    # 上传文件
    try:
        file_name = upload_file(img_bytes)
    except BaseException as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.THIRDERR, errmsg=error_map[RET.THIRDERR])

    # 数据库行为: 修改用户数据
    user.avatar_url = file_name

    # json返回结果  为了让前端可以修改头像, 需要返回新的头像链接
    return jsonify(errno=RET.OK, errmsg=error_map[RET.OK], data=user.to_dict())


# 修改密码
@user_blu.route('/pass_info', methods=['POST'])
@user_login_data
def pass_info():
    user = g.user
    if not user:  # 判断用户是否登录
        return abort(403)
    
    # 获取参数
    old_password = request.json.get('old_password')
    new_password = request.json.get('new_password')
    # 校验参数
    if not all([old_password, new_password]):
        return jsonify(errno=RET.PARAMERR, errmsg=error_map[RET.PARAMERR])
    
    # 先校验旧密码
    if user.check_password(old_password) == False:
        return jsonify(errno=RET.PWDERR, errmsg=error_map[RET.PWDERR])

    # 数据库行为: 修改新密码
    user.password = new_password

    # 返回json
    return jsonify(errno=RET.OK, errmsg=error_map[RET.OK])


# 我的收藏
@user_blu.route('/collection')
@user_login_data
def collection():
    user = g.user
    if not user:  # 判断用户是否登录
        return abort(403)

    # 获取参数
    p = request.args.get("p", 1)
    # 校验参数
    try:
        p = int(p)
    except BaseException as e:
        return abort(403)

    # 数据库行为 分页查询当前用户收藏的所有新闻  按照收藏时间倒序排列
    try:
        pn = user.collection_news.order_by(UserCollection.create_time.desc()).paginate(p, USER_COLLECTION_MAX_NEWS)
    except BaseException as e:
        current_app.logger.error(e)
        return abort(500)

    data = {
        "news_list": [news.to_dict() for news in pn.items],
        "cur_page": pn.page,
        "total_page": pn.pages
    }

    # 后端模板渲染
    return render_template("news/user_collection.html", data=data)


# 显示新闻发布页面/提交新闻数据
@user_blu.route('/news_release', methods=['GET', 'POST'])
@user_login_data
def news_release():
    # 判断用户是否登录
    user = g.user
    if not user:  # 用户未登录
        return abort(403)

    if request.method == 'GET':  # 展示页面
        # 查询所有的分类
        try:
            categories = Category.query.filter(Category.id != 1).all()
        except BaseException as e:
            current_app.logger.error(e)
            return abort(500)

        # 将分类数据传入模板渲染
        return render_template("news/user_news_release.html", categories=categories)

    # POST 提交数据
    # 获取参数
    title = request.form.get("title")
    category_id = request.form.get("category_id")
    digest = request.form.get("digest")
    content = request.form.get("content")
    index_image_file = request.files.get("index_image")

    # 校验参数
    if not all([title, category_id, digest, content, index_image_file]):
        return jsonify(errno=RET.PARAMERR, errmsg=error_map[RET.PARAMERR])
    try:
        category_id = int(category_id)
    except BaseException as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.PARAMERR, errmsg=error_map[RET.PARAMERR])

    # 数据库行为: 生成一条新闻纪录
    news = News()
    news.title = title
    news.category_id = category_id
    news.digest = digest
    news.content = content
    # 上传图片
    try:
        img_bytes = index_image_file.read()
        file_name = upload_file(img_bytes)
        news.index_image_url = QINIU_DOMIN_PREFIX + file_name
    except BaseException as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.THIRDERR, errmsg=error_map[RET.THIRDERR])

    # 设置其他字段
    news.source = "个人发布"  # 发布来源
    news.user_id = user.id  # 作者id
    news.status = 1  # 设置新闻状态  1表示待审核

    # 添加到会话
    db.session.add(news)

    # json返回结果
    return jsonify(errno=RET.OK, errmsg=error_map[RET.OK])


# 新闻列表
@user_blu.route('/news_list')
@user_login_data
def news_list():
    user = g.user
    if not user:  # 判断用户是否登录
        return abort(403)

    # 获取参数
    p = request.args.get("p", 1)
    # 校验参数
    try:
        p = int(p)
    except BaseException as e:
        return abort(403)

    # 数据库行为 分页查询当前用户发布的所有新闻
    try:
        pn = user.news_list.paginate(p, USER_COLLECTION_MAX_NEWS)
    except BaseException as e:
        current_app.logger.error(e)
        return abort(500)

    data = {
        "news_list": [news.to_review_dict() for news in pn.items],
        "cur_page": pn.page,
        "total_page": pn.pages
    }

    # 后端模板渲染
    return render_template("news/user_news_list.html", data=data)


# 我的关注
@user_blu.route('/user_follow')
@user_login_data
def user_follow():
    user = g.user
    if not user:  # 判断用户是否登录
        return abort(403)

    # 获取参数
    p = request.args.get("p", 1)
    # 校验参数
    try:
        p = int(p)
    except BaseException as e:
        return abort(403)

    # 数据库行为 分页查询当前用户关注的所有作者
    try:
        pn = user.followed.paginate(p, USER_FOLLOWED_MAX_COUNT)
    except BaseException as e:
        current_app.logger.error(e)
        return abort(500)

    data = {
        "author_list": [author.to_dict() for author in pn.items],
        "cur_page": pn.page,
        "total_page": pn.pages
    }

    # 后端模板渲染
    return render_template("news/user_follow.html", data=data)